<template>
  <div>
    <!-- 使用 Element UI 的 Popover 组件实现下拉框 -->
    <el-popover
      style="overflow-y: auto; "
      placement="bottom"
      trigger="click">
      <!-- 输入框用于搜索树结构中的节点 -->
      <el-input
        style="margin-bottom: 10px"
        v-model="searchContent"
        clearable
        @clear="searchHandleIptClear"
      >
        <el-button slot="append" icon="el-icon-search" @click="search" size="small" >{{搜索按钮名称}}</el-button>
      </el-input>
      <!-- 树结构展示数据 -->
      <el-tree
        style="width:auto"
        ref="tree"
        :props="props"
        :data="treeData"
        :check-strictly="false"
        @check-change="handleCheckChange"
        show-checkbox
        node-key="id"
        :default-expanded-keys="需要展开的节点keys"
        @node-click="handleNodeClick"
      >

        <span class="custom-tree-node" slot-scope="{ node, data }">
<!--          高亮的部分-->
          <span v-if="需要高亮节点数组 && 需要高亮节点数组.indexOf(data.id) !== -1" :style="高亮样式Style">{{ node.label }}</span>
<!--          不需要高亮的部分-->
          <span v-else>{{ node.label }}</span>
        </span>
      </el-tree>
<!--      输入框 用来显示选中的节点内容-->
      <el-input slot="reference"
                style="width:380px"
                v-model="value.label"
                :placeholder="提示词placeholder"
                clearable
                readonly
                @clear="handleIptClear">
<!--        清除按钮-->
        <i
          slot="suffix"
          @click="IptClearClick"
          style="margin-top: 11px; cursor: pointer"
          class="el-icon-circle-close clear-button"></i>
      </el-input>
    </el-popover>
  </div>
</template>
<script>
export default {
  name: 'TreeSelector', // 组件名称
  /**
   * 属性
   */
  props: {
    // 树结构的数据源
    treeData: {
      type:Array,
      required:true,
      default:()=>[]
    },
    高亮样式Style: {
      type:String,
      default:()=>'background-color: yellow'
    },
    是否区分大小写: {
      type:Boolean,
      default:()=>false
    },
    //搜索按钮名称
    搜索按钮名称:{
      type:String,
      default:()=>'搜索'
    },
    //提示词placeholder
    提示词placeholder:{
      type:String,
      default:()=>'点击选择'
    },
  },
  data() {
    return {
      需要高亮节点数组:[],
      // 已选中的节点keys:[],
      需要展开的节点keys:[],
      searchContent:'',
      props: {
        label: 'label',
        children: 'children'
      },
      value:{id:'', label: ''},
    };
  },
  mounted() {
  },
  methods: {
    //清空搜索框内容事件
    searchHandleIptClear(){
      this.需要高亮节点数组 = []
    },
    高亮模糊查询的节点递归(arr, 父节点){
      if(!arr || arr.length <= 0){
        return
      }
      let that = this
      arr.forEach(item=>{
        let labelTmp = item.label;
        let searchContentTmp = that.searchContent;
        //不区分大小写匹配
        if(that.是否区分大小写 === false){
          labelTmp = labelTmp.toUpperCase()
          searchContentTmp = searchContentTmp.toUpperCase()
        }

        if(labelTmp.indexOf(searchContentTmp) !== -1){
          // console.log('搜索内容匹配', that.searchContent, item, 父节点)
          that.需要高亮节点数组.push(item.id);
          if(父节点){
            that.需要展开的节点keys.push(父节点.id)
          }
        }
        if(item.children){
          that.高亮模糊查询的节点递归(item.children, item)
        }
      })
    },
    /**
     * 点击搜索
     */
    search(){
      let that = this
      //1 过滤所有内容
      that.需要高亮节点数组 = []
      that.需要展开的节点keys = []
      if(that.searchContent){
        this.高亮模糊查询的节点递归(this.treeData, null)
      }
    },
    IptClearClick(event){
      // console.log(event)
      // event.preventDefault();
      this.handleIptClear()
    },
    // 清空输入框内容
    handleIptClear(){
      // console.log('清空输入框内容')
      //清空选中内容
      this.$refs.tree.setCheckedNodes([])
      this.value.id = ''
      this.value.label = ''
    },
    /**
     * 更新被选中的值
     */
    updateCheck(){
      const seltedNodes = this.getCheckedNodes()
      // console.log(seltedNodes)
      const ids = seltedNodes.map(n => n.id)
      const labels = seltedNodes.map(n => n.label)
      this.value.id = ids + ''
      this.value.label = labels + ''
    },
    // checkbox被选中或取消选中
    handleCheckChange(arg1, arg2, arg3){
      // console.log(arg1, arg2, arg3);
      this.updateCheck()
    },
    // 节点被点击
    handleNodeClick(arg1, arg2, arg3){
      // console.log('nodes:', arg1, arg2, arg3)
      this.updateCheck()
    },
    /**
     * 获取已选中的节点集合
     */
    getCheckedNodes(){
      return this.getTreeRef().getCheckedNodes();
    },
    /**
     * 获取已选中的节点keys集合
     */
    getCheckedKeys(){
      return this.getCheckedNodes().map(n => n.id);
    },
    /**
     * 追加设置选中的节点
     * @param arr
     */
    appendSetCheckedKeys(arr){
      let 已选中的keys = this.getCheckedKeys()
      const mergedArray = [...已选中的keys, ...arr];
      const mergedSet = new Set(mergedArray);
      this.setCheckedKeys(Array.from(mergedSet))
    },
    /**
     * 设置选中的节点
     * @param arr
     */
    setCheckedKeys(arr){
      this.getTreeRef().setCheckedKeys(arr)
    },
    /**
     * 获取tree ref
     * https://element.eleme.cn/#/zh-CN/component/tree
     * @returns {HTMLElement}
     */
    getTreeRef(){
      return this.$refs.tree;
    }
  }
};
</script>
